... and pass it to the API that computes new styles.
A special timestamp of 0 means "please don't animate" and is used when
no frame clock is available for a node.
static GtkCssStyle *
gtk_css_node_real_update_style (GtkCssNode *cssnode,
GtkCssChange pending_change,
+ gint64 timestamp,
GtkCssStyle *old_style)
{
if (!gtk_css_style_needs_recreation (old_style, pending_change))
return NULL;
}
+static GdkFrameClock *
+gtk_css_node_real_get_frame_clock (GtkCssNode *cssnode)
+{
+ return NULL;
+}
+
static void
gtk_css_node_class_init (GtkCssNodeClass *klass)
{
klass->create_widget_path = gtk_css_node_real_create_widget_path;
klass->get_widget_path = gtk_css_node_real_get_widget_path;
klass->get_style_provider = gtk_css_node_real_get_style_provider;
+ klass->get_frame_clock = gtk_css_node_real_get_frame_clock;
}
static void
cssnode->visible = TRUE;
}
+static GdkFrameClock *
+gtk_css_node_get_frame_clock_or_null (GtkCssNode *cssnode)
+{
+ while (cssnode->parent)
+ cssnode = cssnode->parent;
+
+ return GTK_CSS_NODE_GET_CLASS (cssnode)->get_frame_clock (cssnode);
+}
+
+static gint64
+gtk_css_node_get_timestamp (GtkCssNode *cssnode)
+{
+ GdkFrameClock *frameclock;
+
+ frameclock = gtk_css_node_get_frame_clock_or_null (cssnode);
+ if (frameclock == NULL)
+ return 0;
+
+ return gdk_frame_clock_get_frame_time (frameclock);
+}
+
static void
gtk_css_node_parent_was_unset (GtkCssNode *node)
{
if (gtk_css_node_get_style_provider_or_null (node) == NULL)
gtk_css_node_invalidate_style_provider (node);
+ gtk_css_node_invalidate (node, GTK_CSS_CHANGE_TIMESTAMP | GTK_CSS_CHANGE_ANIMATIONS);
}
if (parent)
}
}
+static gboolean
+gtk_css_node_needs_new_style (GtkCssNode *cssnode)
+{
+ return cssnode->style_is_invalid;
+}
+
static void
-gtk_css_node_ensure_style (GtkCssNode *cssnode)
+gtk_css_node_ensure_style (GtkCssNode *cssnode,
+ gint64 current_time)
{
GtkCssStyle *new_style;
- if (!cssnode->style_is_invalid)
+ if (!gtk_css_node_needs_new_style (cssnode))
return;
if (cssnode->parent)
- gtk_css_node_ensure_style (cssnode->parent);
+ gtk_css_node_ensure_style (cssnode->parent, current_time);
if (cssnode->previous_sibling)
- gtk_css_node_ensure_style (cssnode->previous_sibling);
+ gtk_css_node_ensure_style (cssnode->previous_sibling, current_time);
new_style = GTK_CSS_NODE_GET_CLASS (cssnode)->update_style (cssnode,
cssnode->pending_changes,
+ current_time,
cssnode->style);
gtk_css_node_propagate_pending_changes (cssnode, new_style != NULL);
GtkCssStyle *
gtk_css_node_get_style (GtkCssNode *cssnode)
{
- gtk_css_node_ensure_style (cssnode);
+ if (gtk_css_node_needs_new_style (cssnode))
+ {
+ gint64 timestamp = gtk_css_node_get_timestamp (cssnode);
+
+ gtk_css_node_ensure_style (cssnode, timestamp);
+ }
return cssnode->style;
}
}
}
+void
+gtk_css_node_invalidate_frame_clock (GtkCssNode *cssnode,
+ gboolean just_timestamp)
+{
+ /* frame clock is handled by the top level */
+ if (cssnode->parent)
+ return;
+
+ if (just_timestamp)
+ gtk_css_node_invalidate (cssnode, GTK_CSS_CHANGE_TIMESTAMP);
+ else
+ gtk_css_node_invalidate (cssnode, GTK_CSS_CHANGE_TIMESTAMP | GTK_CSS_CHANGE_ANIMATIONS);
+}
+
void
gtk_css_node_invalidate (GtkCssNode *cssnode,
GtkCssChange change)
const GtkWidgetPath * (* get_widget_path) (GtkCssNode *cssnode);
/* get style provider to use or NULL to use parent's */
GtkStyleProviderPrivate *(* get_style_provider) (GtkCssNode *cssnode);
+ /* get frame clock or NULL (only relevant for root node) */
+ GdkFrameClock * (* get_frame_clock) (GtkCssNode *cssnode);
GtkCssStyle * (* update_style) (GtkCssNode *cssnode,
GtkCssChange pending_changes,
+ gint64 timestamp,
GtkCssStyle *old_style);
void (* invalidate) (GtkCssNode *node);
void (* queue_validate) (GtkCssNode *node);
void gtk_css_node_invalidate_style_provider
(GtkCssNode *cssnode);
+void gtk_css_node_invalidate_frame_clock
+ (GtkCssNode *cssnode,
+ gboolean just_timestamp);
void gtk_css_node_invalidate (GtkCssNode *cssnode,
GtkCssChange change);
void gtk_css_node_validate (GtkCssNode *cssnode,
{ GTK_CSS_CHANGE_POSITION, GTK_CSS_CHANGE_SIBLING_POSITION },
{ GTK_CSS_CHANGE_STATE, GTK_CSS_CHANGE_SIBLING_STATE },
{ GTK_CSS_CHANGE_SOURCE, 0 },
- { GTK_CSS_CHANGE_ANIMATE, 0 },
{ GTK_CSS_CHANGE_PARENT_STYLE, 0 }
};
{ GTK_CSS_CHANGE_SIBLING_POSITION, GTK_CSS_CHANGE_PARENT_SIBLING_POSITION },
{ GTK_CSS_CHANGE_SIBLING_STATE, GTK_CSS_CHANGE_PARENT_SIBLING_STATE },
{ GTK_CSS_CHANGE_SOURCE, 0 },
- { GTK_CSS_CHANGE_ANIMATE, 0 },
{ GTK_CSS_CHANGE_PARENT_STYLE, 0 }
};
GTK_CSS_CHANGE_PARENT_SIBLING_STATE = (1 << 15),
/* add more */
GTK_CSS_CHANGE_SOURCE = (1 << 16),
- GTK_CSS_CHANGE_ANIMATE = (1 << 17),
- GTK_CSS_CHANGE_PARENT_STYLE = (1 << 18),
+ GTK_CSS_CHANGE_PARENT_STYLE = (1 << 17),
+ GTK_CSS_CHANGE_TIMESTAMP = (1 << 18),
+ GTK_CSS_CHANGE_ANIMATIONS = (1 << 19),
GTK_CSS_CHANGE_RESERVED_BIT = (1 << 31) /* Used internally in gtkcssselector.c */
} GtkCssChange;
static GtkCssStyle *
gtk_css_widget_node_update_style (GtkCssNode *cssnode,
GtkCssChange pending_change,
+ gint64 timestamp,
GtkCssStyle *old_style)
{
if (old_style == NULL)
- return GTK_CSS_NODE_CLASS (gtk_css_widget_node_parent_class)->update_style (cssnode, pending_change, old_style);
+ return GTK_CSS_NODE_CLASS (gtk_css_widget_node_parent_class)->update_style (cssnode, pending_change, timestamp, old_style);
return NULL;
}
+static gboolean
+gtk_css_widget_node_queue_callback (GtkWidget *widget,
+ GdkFrameClock *frame_clock,
+ gpointer user_data)
+{
+ GtkCssNode *node = user_data;
+
+ gtk_css_node_invalidate_frame_clock (node, TRUE);
+ _gtk_container_queue_restyle (GTK_CONTAINER (widget));
+
+ return G_SOURCE_CONTINUE;
+}
+
static void
gtk_css_widget_node_queue_validate (GtkCssNode *node)
{
G_GNUC_BEGIN_IGNORE_DEPRECATIONS
if (GTK_IS_RESIZE_CONTAINER (widget_node->widget))
widget_node->validate_cb_id = gtk_widget_add_tick_callback (widget_node->widget,
- (GtkTickCallback) _gtk_container_queue_restyle,
- NULL,
+ gtk_css_widget_node_queue_callback,
+ node,
NULL);
G_GNUC_END_IGNORE_DEPRECATIONS
}
new_static_style = validate_static_style (node, static_style, change);
- if (new_static_style != static_style)
+ if (new_static_style != static_style || (change & GTK_CSS_CHANGE_ANIMATIONS))
{
GtkCssNode *parent = gtk_css_node_get_parent (node);
new_style = gtk_css_animated_style_new (new_static_style,
g_object_unref (new_static_style);
}
- else if (GTK_IS_CSS_ANIMATED_STYLE (style))
+ else if (GTK_IS_CSS_ANIMATED_STYLE (style) && (change & GTK_CSS_CHANGE_TIMESTAMP))
{
new_style = gtk_css_animated_style_new_advance (GTK_CSS_ANIMATED_STYLE (style),
static_style,
return gtk_style_context_get_style_provider (gtk_widget_get_style_context (widget_node->widget));
}
+static GdkFrameClock *
+gtk_css_widget_node_get_frame_clock (GtkCssNode *node)
+{
+ GtkCssWidgetNode *widget_node = GTK_CSS_WIDGET_NODE (node);
+
+ if (widget_node->widget == NULL)
+ return NULL;
+
+ return gtk_widget_get_frame_clock (widget_node->widget);
+}
+
static void
gtk_css_widget_node_class_init (GtkCssWidgetNodeClass *klass)
{
node_class->create_widget_path = gtk_css_widget_node_create_widget_path;
node_class->get_widget_path = gtk_css_widget_node_get_widget_path;
node_class->get_style_provider = gtk_css_widget_node_get_style_provider;
+ node_class->get_frame_clock = gtk_css_widget_node_get_frame_clock;
}
static void
gdk_frame_clock_begin_updating (frame_clock);
}
+ gtk_css_node_invalidate_frame_clock (priv->cssnode, FALSE);
+
if (priv->context)
gtk_style_context_set_frame_clock (priv->context, frame_clock);
}
if (GTK_IS_CONTAINER (widget))
_gtk_container_stop_idle_sizer (GTK_CONTAINER (widget));
+ gtk_css_node_invalidate_frame_clock (priv->cssnode, FALSE);
+
if (priv->clock_tick_id)
{
g_signal_handler_disconnect (frame_clock, priv->clock_tick_id);